02 - Introduction to ROS

Robotics I

Poznan University of Technology, Institute of Robotics and Machine Intelligence

Laboratory 2: Introduction to Robot Operating System

Goals

Resources

Robot Operating System


Source: https://www.ros.org

The Robot Operating System (ROS) is a set of software libraries and tools that help you build robot applications. From drivers to state-of-the-art algorithms, and with powerful developer tools, ROS has what you need for your next robotics project. And it’s all open source.
https://www.ros.org/


The Robot Operating System (ROS) was first released in 2007 as ROS 1 for advanced research projects. However, ROS 1 was limited by weaknesses related to message access security and lack of adaptation to the needs of real-time systems. Therefore, the second generation, ROS 2, was redesigned to address these challenges, with the first release in 2017. Despite this change, the main concepts remain the same:

During the labs we will use the latest stable version of ROS 2 - Jazzy Jalisco.

Basic concepts

Workspace

A ROS 2 workspace is a directory containing multiple packages with build configurations and dependencies, e.g. for a particular robot. You can have many different workspaces on one machine (e.g. ros2_robotA_ws, ros2_robotB_ws). Usually ROS 2 workspaces are structured as follows:

ros2_ws
├── build - contains files for the build process, so that only new changes can be included in the re-build
├── install - location where packages are installed
├── log - contains the log of the build process
└── src - directory containing developer's ROS 2 packages

The ROS 2 environment is built using colcon, with the following command:

colcon build

If necessary, it is possible to build a single package:

colcon build --packages-select <PACKAGE NAME>

The first and required step before working with the built ROS 2 environment is to set the environment variables (source), in order to access the packages in a particular terminal.

In the ~/ros2_ws directory, call the command:

source install/setup.bash

At the first launch, if the environment is not yet built, it may be necessary to activate the global ROS 2 variables using the below command. In our case, ROS 2 distribution is jazzy.

source /opt/ros/$ROS_DISTRO/setup.bash

Note: Every time you open a terminal window and want to work in a specific environment, run the above command in the appropriate directory! It should be called in the root directory of the ROS 2 environment (e.g. ~/ros2_ws).

Package

A package is essential container that organize nodes and structurize source code, increasing its modularity. It enables easy management and integration of nodes, executables, and other resources within the ROS 2 ecosystem by specifying metadata and build instructions.

Example package structure in Python is as below:

my_package/
├── my_package        - directory containing the nodes
│   └── __init__.py
├── package.xml       - contains information about the package and its dependencies
├── resource
│   └── my_package    - directory required to locate the package
├── setup.cfg         - holds information about the executable files
├── setup.py          - contains instructions for installing the package
└── test              - contains scripts for automatic testing

The organization of packages is done by using ament, an evolution of catkin known from ROS 1. To create a template package in C++ and Python simply call the following functions: - C++

ros2 pkg create --build-type ament_cmake <PACKAGE NAME>
ros2 pkg create --build-type ament_python <PACKAGE NAME>

It is also possible to indicate dependencies with the --dependencies argument and to create a node at the same time via the additional --node-name argument.

Package Dependencies Management

A tool that greatly improves the work of a ROS 2 developer is rosdep. It allows automatic installation of dependencies (packages, libraries) for all packages in a given environment. Thanks to the fact that dependencies are defined in the package.xml file, there is no need to install them manually.

Many ready-made packages are available in the ROS 2 repository - rosdistro, any user has the possibility to add his package via pull request.

To use rosdep, run the following commands in the ~/ros2_ws directory:

sudo rosdep init
rosdep update
rosdep install --from-paths src -y --ignore-src --rosdistro jazzy

These commands initialize rosdep and then update the local indexes from the rosdistro package database. The last command installs the dependencies. The --from-paths src argument tells it to look for package.xml file in the src directory, -y causes console messages to be accepted automatically, and --ignore-src omits packages in the src directory from the installation (since we build them).

Graph concept

A ROS graph is a network of nodes that process data. It includes all nodes and the communication links between them. A ROS 2 graph contains:

To visualize the current state of the graph, changing nodes and topics, as well as the connections between them, one can use rqt_graph tool:

rqt_graph
Example graph visualization:


Source: https://docs.ros.org

Node

A node represents an individual process in ROS 2. Nodes perform computations, publish or subscribe to topics, and communicate with each other to perform specific tasks within a robotic system.

Nodes are started using the command:

ros2 run <PACKAGE NAME> <NODE NAME>

To get the current list of nodes call:

ros2 node list

To get node information:

ros2 node info <NODE NAME>
Launch files

For more complex ROS 2 applications it is useful to group nodes so that they can be run together. This can be done using launch files, which can define and configure not only the execution of multiple nodes, but also runtime parameters, and actions, allowing automated system startup and management. In ROS 2, launch files can be written in Python, XML or YAML files, with the following extensions .xml, .py or .yaml. See the “Using Python, XML, and YAML for ROS 2 Launch Files” article for more information.

To invoke an existing launch file, use the command:

ros2 run <PACKAGE NAME> <LAUNCH NAME>

Message

Messages in ROS 2 are data structures used for communication between nodes to exchange information such as sensor data, commands, and states in a standardized format. There are many message types natively available in ROS 2 and they can contain different types of information (e.g. location, orientation, camera image). Examples of standard message types include:

It is also possible to create your own message type - Creating custom msg and srv files.

To check the message interface definition, use the command:

ros2 interface show <MESSAGE TYPE>

Topic

In ROS 2, topics are uniquely named communication channels that facilitate the asynchronous exchange of messages between nodes, allowing data to be published and subscribed to by different components of the system. A topic is created when a node begins publishing a particular type of message. A single node can publish messages to multiple topics and subscribe to messages from multiple topics. At the same time, a single topic can have multiple publishers and subscribers.


Source: https://docs.ros.org/en/jazzy/Tutorials/Beginner-CLI-Tools/Understanding-ROS2-Topics/Understanding-ROS2-Topics.html

To see the current list of topics, use the command below. By adding the `-t’ parameter, it is possible to display topics with their associated message types:

ros2 topic list

To read messages from a topic, simply use:

ros2 topic echo <TOPIC NAME>

To get information about a specific topic, as well as the type of message being exchanged, use the command:

ros2 topic info <TOPIC NAME>

It is also possible to post messages to a topic from the terminal:

ros2 topic pub <TOPIC NAME> <MESSAGE TYPE> "message_data"

Example:

ros2 topic pub --once /turtle1/cmd_vel geometry_msgs/msg/Twist "{linear: {x: 2.0, y: 0.0, z: 0.0}, angular: {x: 0.0, y: 0.0, z: 1.8}}"

To publish a message once, you can use the `–once’ argument.

To read the frequency with which data is published on a topic, use the hz argument:

ros2 topic hz <TOPIC NAME>

Message visualization

RViz is a 3D visualization tool that allows users to display information from topics, interactively view and analyze sensor data, robot states, and planning information in real time. In ROS 2, RViz is launched with the command:

rviz2

To add a new topic to the RViz visualizer, click the Add button (1). Then, in the pop-up window select By topic tab (2) and select the topic you are looking for from the list (3).

Dynamic simulation

Gazebo in ROS 2 is a simulation tool that allows testing and development of robots in a virtual environment, offering realistic physics, sensor models, interaction, and 3D visualization. Gazebo can be launched with the command:

gz sim


Tasks

Note 1: Everything we do today should be done inside the container! To attach another terminal to a running container use the command docker exec -it <CONTAINER NAME> bash.

Note 2: Every time you open a terminal window and want to work with ROS 2 you need to source environmental variables using source /opt/ros/$ROS_DISTRO/setup.bash or source install/setup.bash if you work in a specific ROS 2 workspace (e.g. ~/ros2_ws)!

  1. Pull osrf/ros:jazzy-desktop-full docker image and create container using docker_run.sh script.
docker_run.sh
IMAGE_NAME="" # <DOCKER IMAGE REPOSITORY>:<DOCKER IMAGE TAG>
CONTAINER_NAME="" # student ID number

xhost +local:root

XAUTH=/tmp/.docker.xauth
if [ ! -f $XAUTH ]
then
    xauth_list=$(xauth nlist :0 | sed -e 's/^..../ffff/')
    if [ ! -z "$xauth_list" ]
    then
        echo $xauth_list | xauth -f $XAUTH nmerge -
    else
        touch $XAUTH
    fi
    chmod a+r $XAUTH
fi

docker stop $CONTAINER_NAME || true && docker rm $CONTAINER_NAME || true

docker run -it \
    --env="ROS_AUTOMATIC_DISCOVERY_RANGE=LOCALHOST" \
    --env="DISPLAY=$DISPLAY" \
    --env="QT_X11_NO_MITSHM=1" \
    --volume="/tmp/.X11-unix:/tmp/.X11-unix:rw" \
    --env="XAUTHORITY=$XAUTH" \
    --volume="$XAUTH:$XAUTH" \
    --privileged \
    --network=host \
    --name="$CONTAINER_NAME" \
    $IMAGE_NAME \
    /bin/bash
  1. Create workspace and src directory (e.g. ~/ros2_ws/src).
  2. Install opencv_cam that supports camera under ROS 2 to test how the camera operation works. To do so, follow these steps:
git clone https://github.com/clydemcqueen/opencv_cam.git
git clone https://github.com/ptrmu/ros2_shared.git
colcon build

source install/setup.bash
ros2 run opencv_cam opencv_cam_main
ros2 run opencv_cam opencv_cam_main --ros-args --params-file opencv_cam_params.yaml
  1. Using rqt_image_view node preview camera stream. Repeat this step using the RViz message visualization tool. In both image visualization tools take a screenshot and upload it to the eKursy platform.
  2. Using rqt_graph visualize ROS 2 graph for the previous task. Check how the graph changes when the messages visualizer is running, which subscribes to the published topic.
  3. In the src directory of ROS 2 workspace create a sample C++-based package and inspect it. To view the project structure, use the tree command (use apt install tree to install it if it is not available). Then, create a sample Python-based package and compare it with the previous one. Shortly, in 2-3 sentences, describe key differences in project structures (C++-based and Python-based) and consider what are the potential applications of each approach. Write your answer in a text file and upload it to eKursy.
  4. Following the instructions in ROS + Gazebo Sim demos, simulate air pressure, IMU, RGBD camera, depth camera and GPU lidar. For each sensor, use the command line tools to check and note which messages it publishes and at what frequency. In addition, for RGBD camera, Depth camera, or GPU lidar, add additional objects to the simulator and check how they affect the measurements. Take a screenshot from RViz with topics preview and add it to eKursy. Do the same for a text file describing the published topics of the above five sensors..